-
-
Notifications
You must be signed in to change notification settings - Fork 452
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Dynamic scaling & Graph meter coloring (new) #714
base: main
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just a quick note:
Please split the new functions for PrevPowerOfTwo and PopCount into their own commit. Also try to utilize configure.ac
to check for them and provide static inline
implementations directly in the header (XUtils.h
or Macros.h
both should work).
There's an example how to check for specific compiler features in configure.ac
to determine whether certain warning flags are supported. You should be able to work based on that example there.
Having the caption above the unit makes more sense IMHO. |
|
k. :)
As inline typically just a hint to the compiler it can still decide to ignore it. Also most recent optimizers do code folding anyway, thus would remove the parts imported from different code units upon linking. Also there's some more magic going on with linkers nowadays, thus I doubt this is a real issue with recent compilers. Compilers can still opt to ignore the inline anytime and will happily do if the code becomes too complex or too wasteful when inlining (unless forced to).
Similar argument with popcount: For any reasonable recent compiler the libgcc function will likely be optimized enough (and likely even use SSE2 when the system has it available), thus only the first call to such a function will receive a slowdown. Also most architectures by now have a NB: Modern compilers are usually better in optimizing your code, than you are. Better optimize the algorithm by replacing it with a better one than trying to be smart … |
If I have not read the disassembly of |
2cdcb94
to
60ca4f2
Compare
Did some tests with the Similar effects are likely visible with the |
prevPowerOf2(0) should (in theory) output 0, not 1. |
14e63c2
to
e0b6f94
Compare
@Explorer09 Why write everything onto screen directly instead of using |
It would break dynamic scaling. Look at Load average meter for example, the whole graph needs to be redrawn when the scale (or unit) changes. |
Sure, but those situations can be tracked/detected … |
@BenBE The second reason for not using a RichString buffer is the use may change color scheme of htop at runtime, and the RichString buffer is not designed to reflect color scheme changes. Detecting when to clean the buffers like the cases above is too much work, and so I consider it doesn't worth the trouble. |
On the other hand, re-using existing code makes maintaining the code easier once it got merged. And in the current state I'm not too convinced that these 650 LOC of new code are in a state that makes maintaining them easy enough to justify the complexity in this code. Just skimming over I saw several routines, that are justifiably quite long. And I also understand you are hesitant to refactor major parts of that code after that much effort that went in. I had some PRs of mine that needed refactoring and bug fixing for about half a year until I could finally merge them. And even then there were small issues that needed to be addressed. So given the amount of code in this PR it should be one priority to ease maintaining the code and another one to have it working. Even if this sound's like I'm a killjoy here both are requirements to proceed with merging (this) new code. |
cd46429
to
3485b0d
Compare
First of all: Thanks a lot for picking up on this again! The new code works and it gives even more precision. Either way I am happy if the makes its way into master. |
0760ab7
to
24bd56e
Compare
This pull request introduces 2 alerts when merging 24bd56e into ed82ce6 - view on LGTM.com new alerts:
|
aa10e7c
to
17e8f97
Compare
9cb1c10
to
cee45a7
Compare
@eworm-de I took my time to revise this patchset.
|
cee45a7
to
9cdf3d8
Compare
This seems to accidentally introduce patches from #1533 … |
9cdf3d8
to
66fba18
Compare
240f9e6
to
76b6f29
Compare
76b6f29
to
601e566
Compare
Meter.c
Outdated
double total; | ||
if (isPercentChart) { | ||
total = MAXIMUM(this->total, sum); | ||
} else { | ||
int scaleExp = 0; | ||
(void)frexp(sum, &scaleExp); | ||
if (scaleExp < 0) { | ||
scaleExp = 0; | ||
} | ||
// In IEEE 754 binary64 (DBL_MAX_EXP == 1024, DBL_MAX_10_EXP == 308), | ||
// "scaleExp" never overflows. | ||
assert(DBL_MAX_10_EXP < 9864); | ||
assert(scaleExp <= INT16_MAX); | ||
valueStart[0].scaleExp = (int16_t)scaleExp; | ||
total = ldexp(1.0, scaleExp); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have several issues with this block of code:
- The uninitialized variable declaration. It's not wrong, as both code paths eventually initialize it, but it is prone to error and thus I'd rather like to avoid it.
- Also a blank above couldn't hurt to separate this block.
- The use of magic numbers in the else block, which aren't even commonly encountered, is a bit concerning too.
- The else block may need a quick comment for what it tries to do. Good to have the description for the
asserts
, but that's just half of the story. - Do you think we can use the true-branch to init total and overwrite it with the false-branch if necessary? The compiler should see this dead store and eliminate the initial assignment if necessary. This also avoids the potential for leaving
total
uninitialized.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you think we can use the true-branch to init total and overwrite it with the false-branch if necessary? The compiler should see this dead store and eliminate the initial assignment if necessary. This also avoids the potential for leaving
total
uninitialized.
I like the idea and it looks like it can save one line of code.
The use of magic numbers in the else block, which aren't even commonly encountered, is a bit concerning too.
The assert is actually redundant if htop only supports IEEE 754 floating points. I have explained what the magic numbers do (DBL_MAX_EXP == 1024, DBL_MAX_10_EXP == 308
), but I used the 9864 for a lax checking, which might have confused you. (2^32768 ≈ 10^9864)
I think I could remove that assert to reduce confusion. Or should I assert(DBL_MAX_10_EXP <= 308)
instead?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Without the hint on 2^32768 ≈ 10^9864
the 9864
is hard to understand. And with the comment above, I'd probably prefer to use the assert(DBL_MAX_10_EXP <= 308)
for the check.
// Convert GraphDataCell.c.details bit representation to Unicode braille | ||
// dot ordering. | ||
// (Bit0) a b (Bit3) From: h g f e d c b a (binary) | ||
// (Bit1) c d (Bit4) | | | X X | | ||
// (Bit2) e f (Bit5) | | | | \ / | | | ||
// (Bit6) g h (Bit7) | | | | X | | | ||
// To: 0x2800 + h g f d b e c a | ||
// Braille Patterns [U+2800, U+28FF] in UTF-8: [E2 A0 80, E2 A3 BF] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This diagram should be somewhat more obvious in the code; eg. as a comment for the details
field of the structure declaration.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I had an idea of presenting the details
values not as Unicode braille characters, but as Unicode "block octant" characters (U+1CD00 - U+1CDE5). So the "transformation to braille code point" diagram would go here.
I can add a comment to explain what the details
field does in the structure definition, however.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The diagram as is is somewhat easy to follow. It only is a bit burried within the code.
33626eb
to
abdaf29
Compare
abdaf29
to
8d617ce
Compare
This suppresses a "-Wshorten-64-to-32" warning in Clang 19.
This property distinguishes meters that have a (relatively) fixed "total" value and meters that do not have a definite maximum value. The former meters would be drawn as a "100% stacked bar" or "graph", and the latter could have their "total" values updated automatically in bar meter mode. This commit is a prerequisite for the feature "Graph meter dynamic scaling and percent graph drawing". Signed-off-by: Kang-Che Sung <[email protected]>
If "isPercentChart" of a meter is false, update its "total" value automatically in the bar meter mode. The "total" value is capped to DBL_MAX in order to ensure the division never produces NaN. The newly introduced Meter_computeSum() function will be reused by the feature "Graph meter dynamic scaling and percent graph drawing". Signed-off-by: Kang-Che Sung <[email protected]>
Implement dynamic scaling for Graph meter mode, and separate it from "100% graph" drawing. This is controlled by the "isPercentChart" property of a MeterClass. If "isPercentChart" is true, the meter would be drawn as a "100% graph". Graph meters now expect the "total" value may change, and the newly changed "total" value no longer affects the percent graph drawings of earlier meter values. If "isPercentChart" is false, the meter would be drawn with a dynamic scale. Signed-off-by: Kang-Che Sung <[email protected]>
Round the graph meter's dynamic scale to a power of two and print the graph scale. For a percent graph, a "%" character is printed in place of the scale. Signed-off-by: Kang-Che Sung <[email protected]>
This is a code quality change that avoids dependency on the hard-coded GRAPH_HEIGHT in GraphMeterMode_draw(). This doesn't enable variable graph heights per meter, but it makes room for implementing such feature. Signed-off-by: Kang-Che Sung <[email protected]>
This is a prerequisite for the feature "Graph meter coloring (with GraphData structure rework)". powerOf2Floor() will utilize __builtin_clz() or stdc_bit_floor_ui() (__builtin_clz() is preferred) if either is supported. popCount8() will utilize ARM NEON instructions and x86 POPCNT instruction if the machine supports either of them. I am not adopting the C23 standard interface stdc_count_ones_uc() yet, as I am not sure C libraries would implement it as fast as our version. Signed-off-by: Kang-Che Sung <[email protected]>
Rewrite the entire graph meter drawing code to support color graph drawing in addition to the dynamic scaling (both can work together because of the new GraphData structure design). The colors of a graph are based on the percentages of item values of the meter. The rounding differences of each terminal character are addressed through the different numbers of braille dots. Due to low resolution of the character terminal, the rasterized colors may not look nice, but better than nothing. :) The new GraphData structure design has two technical limitations: * The height of a graph meter now has a limit of 8191 (terminal rows). * If dynamic scaling is used, the "total" value or scale of a graph now has to align to a power of 2. The code is designed with the anticipation that the graph height may change at runtime. No UI or option has been implemented for that yet. Signed-off-by: Kang-Che Sung <[email protected]>
Specifically 'PIXPERROW_*' and 'GraphMeterMode_dots*' constants. They were commented out rather than removed in the previous commit (for ease of code reviewing). Now this commit removes the constant defines for good.
8d617ce
to
a0a929e
Compare
This pull request implements dynamic scaling support and a "stacked" color for Graph meters. In addition, the Bar meter drawing routine is slightly improved for dynamically expanding the "total" value when needed.
This code would replace the current, one-color graph drawing algorithm so that all Graph meters are drawn with colors.
The data structures and algorithms for the color graphs and dynamically scaling are brand new, totally reworked from #129 and the colors are intended to represent the meter data as close as possible. It still uses braille dots to display the graph, but they now have finer details (I think).
Caveats
double
(floating point) data type. It now stores the precomputed colors and braille dots instead, and the data are specific to a graph height setting.CRT.h
would become unused.Technical details
The following are potential features that can be implemented after this pull request, but I doubt if they are good ideas or if I have the ability to implement them:
%
,32K
,16M
, etc.), separate from the caption color. I was considering making the text blue to separate it from the cyan text of the caption.htoprc
file to specify the graph height. The settings UI would be the next step to implement.|#*@$%&
) can be reused for this.